# SPDX-FileCopyrightText: 2022-present deepset GmbH <info@deepset.ai>
#
# SPDX-License-Identifier: Apache-2.0

import logging
from typing import Any, Optional, Union

import arrow
import pytest
from jinja2 import TemplateSyntaxError

from haystack import component
from haystack.components.builders.chat_prompt_builder import ChatPromptBuilder
from haystack.core.pipeline.pipeline import Pipeline
from haystack.dataclasses.chat_message import ChatMessage, ImageContent, ReasoningContent
from haystack.dataclasses.document import Document


class TestChatPromptBuilder:
    def test_init(self):
        builder = ChatPromptBuilder(
            template=[
                ChatMessage.from_user("This is a {{ variable }}"),
                ChatMessage.from_system("This is a {{ variable2 }}"),
            ]
        )
        assert builder.required_variables == []
        assert builder.template[0].text == "This is a {{ variable }}"
        assert builder.template[1].text == "This is a {{ variable2 }}"
        assert builder._variables is None
        assert builder._required_variables is None

        # we have inputs that contain: template, template_variables + inferred variables
        inputs = builder.__haystack_input__._sockets_dict
        assert set(inputs.keys()) == {"template", "template_variables", "variable", "variable2"}
        assert inputs["template"].type == Optional[Union[list[ChatMessage], str]]
        assert inputs["template_variables"].type == Optional[dict[str, Any]]
        assert inputs["variable"].type == Any
        assert inputs["variable2"].type == Any

        # response is always prompt
        outputs = builder.__haystack_output__._sockets_dict
        assert set(outputs.keys()) == {"prompt"}
        assert outputs["prompt"].type == list[ChatMessage]

    def test_init_without_template(self):
        variables = ["var1", "var2"]
        builder = ChatPromptBuilder(variables=variables)
        assert builder.template is None
        assert builder.required_variables == []
        assert builder._variables == variables
        assert builder._required_variables is None

        # we have inputs that contain: template, template_variables + variables
        inputs = builder.__haystack_input__._sockets_dict
        assert set(inputs.keys()) == {"template", "template_variables", "var1", "var2"}
        assert inputs["template"].type == Optional[Union[list[ChatMessage], str]]
        assert inputs["template_variables"].type == Optional[dict[str, Any]]
        assert inputs["var1"].type == Any
        assert inputs["var2"].type == Any

        # response is always prompt
        outputs = builder.__haystack_output__._sockets_dict
        assert set(outputs.keys()) == {"prompt"}
        assert outputs["prompt"].type == list[ChatMessage]

    def test_init_with_required_variables(self):
        builder = ChatPromptBuilder(
            template=[ChatMessage.from_user("This is a {{ variable }}")], required_variables=["variable"]
        )
        assert builder.required_variables == ["variable"]
        assert builder.template[0].text == "This is a {{ variable }}"
        assert builder._variables is None
        assert builder._required_variables == ["variable"]

        # we have inputs that contain: template, template_variables + inferred variables
        inputs = builder.__haystack_input__._sockets_dict
        assert set(inputs.keys()) == {"template", "template_variables", "variable"}
        assert inputs["template"].type == Optional[Union[list[ChatMessage], str]]
        assert inputs["template_variables"].type == Optional[dict[str, Any]]
        assert inputs["variable"].type == Any

        # response is always prompt
        outputs = builder.__haystack_output__._sockets_dict
        assert set(outputs.keys()) == {"prompt"}
        assert outputs["prompt"].type == list[ChatMessage]

    def test_init_with_custom_variables(self):
        variables = ["var1", "var2", "var3"]
        template = [ChatMessage.from_user("Hello, {{ var1 }}, {{ var2 }}!")]
        builder = ChatPromptBuilder(template=template, variables=variables)
        assert builder.required_variables == []
        assert builder._variables == variables
        assert builder.template[0].text == "Hello, {{ var1 }}, {{ var2 }}!"
        assert builder._required_variables is None

        # we have inputs that contain: template, template_variables + variables
        inputs = builder.__haystack_input__._sockets_dict
        assert set(inputs.keys()) == {"template", "template_variables", "var1", "var2", "var3"}
        assert inputs["template"].type == Optional[Union[list[ChatMessage], str]]
        assert inputs["template_variables"].type == Optional[dict[str, Any]]
        assert inputs["var1"].type == Any
        assert inputs["var2"].type == Any
        assert inputs["var3"].type == Any

        # response is always prompt
        outputs = builder.__haystack_output__._sockets_dict
        assert set(outputs.keys()) == {"prompt"}
        assert outputs["prompt"].type == list[ChatMessage]

    def test_run(self):
        builder = ChatPromptBuilder(template=[ChatMessage.from_user("This is a {{ variable }}")])
        res = builder.run(variable="test")
        assert res == {"prompt": [ChatMessage.from_user("This is a test")]}

    def test_run_template_variable(self):
        builder = ChatPromptBuilder(template=[ChatMessage.from_user("This is a {{ variable }}")])
        res = builder.run(template_variables={"variable": "test"})
        assert res == {"prompt": [ChatMessage.from_user("This is a test")]}

    def test_run_template_variable_overrides_variable(self):
        builder = ChatPromptBuilder(template=[ChatMessage.from_user("This is a {{ variable }}")])
        res = builder.run(template_variables={"variable": "test_from_template_var"}, variable="test")
        assert res == {"prompt": [ChatMessage.from_user("This is a test_from_template_var")]}

    def test_run_without_input(self):
        builder = ChatPromptBuilder(template=[ChatMessage.from_user("This is a template without input")])
        res = builder.run()
        assert res == {"prompt": [ChatMessage.from_user("This is a template without input")]}

    def test_run_with_missing_input(self):
        builder = ChatPromptBuilder(template=[ChatMessage.from_user("This is a {{ variable }}")])
        res = builder.run()
        assert res == {"prompt": [ChatMessage.from_user("This is a ")]}

    def test_run_with_missing_required_input(self):
        builder = ChatPromptBuilder(
            template=[ChatMessage.from_user("This is a {{ foo }}, not a {{ bar }}")], required_variables=["foo", "bar"]
        )
        with pytest.raises(ValueError, match="foo"):
            builder.run(bar="bar")
        with pytest.raises(ValueError, match="bar"):
            builder.run(foo="foo")
        with pytest.raises(ValueError, match="foo, bar"):
            builder.run()

    def test_run_with_missing_required_input_using_star(self):
        builder = ChatPromptBuilder(
            template=[ChatMessage.from_user("This is a {{ foo }}, not a {{ bar }}")], required_variables="*"
        )
        with pytest.raises(ValueError, match="foo"):
            builder.run(bar="bar")
        with pytest.raises(ValueError, match="bar"):
            builder.run(foo="foo")
        with pytest.raises(ValueError, match="bar, foo"):
            builder.run()

    def test_run_with_variables(self):
        variables = ["var1", "var2", "var3"]
        template = [ChatMessage.from_user("Hello, {{ name }}! {{ var1 }}")]

        builder = ChatPromptBuilder(template=template, variables=variables)

        template_variables = {"name": "John"}
        expected_result = {"prompt": [ChatMessage.from_user("Hello, John! How are you?")]}

        assert builder.run(template_variables=template_variables, var1="How are you?") == expected_result

    def test_run_with_variables_and_runtime_template(self):
        variables = ["var1", "var2", "var3"]

        builder = ChatPromptBuilder(variables=variables)

        template = [ChatMessage.from_user("Hello, {{ name }}! {{ var1 }}")]
        template_variables = {"name": "John"}
        expected_result = {"prompt": [ChatMessage.from_user("Hello, John! How are you?")]}

        assert (
            builder.run(template=template, template_variables=template_variables, var1="How are you?")
            == expected_result
        )

    def test_run_overwriting_default_template(self):
        default_template = [ChatMessage.from_user("Hello, {{ name }}!")]

        builder = ChatPromptBuilder(template=default_template)

        template = [ChatMessage.from_user("Hello, {{ var1 }}{{ name }}!")]
        expected_result = {"prompt": [ChatMessage.from_user("Hello, John!")]}

        assert builder.run(template, name="John") == expected_result

    def test_run_overwriting_default_template_with_template_variables(self):
        default_template = [ChatMessage.from_user("Hello, {{ name }}!")]

        builder = ChatPromptBuilder(template=default_template)

        template = [ChatMessage.from_user("Hello, {{ var1 }} {{ name }}!")]
        template_variables = {"var1": "Big"}
        expected_result = {"prompt": [ChatMessage.from_user("Hello, Big John!")]}

        assert builder.run(template, template_variables, name="John") == expected_result

    def test_run_overwriting_default_template_with_variables(self):
        variables = ["var1", "var2", "name"]
        default_template = [ChatMessage.from_user("Hello, {{ name }}!")]

        builder = ChatPromptBuilder(template=default_template, variables=variables)

        template = [ChatMessage.from_user("Hello, {{ var1 }} {{ name }}!")]
        expected_result = {"prompt": [ChatMessage.from_user("Hello, Big John!")]}

        assert builder.run(template, name="John", var1="Big") == expected_result

    def test_run_with_meta(self):
        """
        Test that the ChatPromptBuilder correctly handles meta data.
        It should render the message and copy the meta data from the original message.
        """
        m = ChatMessage.from_user("This is a {{ variable }}")
        m.meta["meta_field"] = "meta_value"
        builder = ChatPromptBuilder(template=[m])
        res = builder.run(variable="test")

        expected_msg = ChatMessage.from_user("This is a test")
        expected_msg.meta["meta_field"] = "meta_value"
        assert res == {"prompt": [expected_msg]}

    def test_run_with_invalid_template(self):
        builder = ChatPromptBuilder()

        template = [ChatMessage.from_user("Hello, {{ name }!")]
        template_variables = {"name": "John"}
        with pytest.raises(TemplateSyntaxError):
            builder.run(template, template_variables)

    def test_init_with_invalid_template(self):
        template = [ChatMessage.from_user("Hello, {{ name }!")]
        with pytest.raises(TemplateSyntaxError):
            ChatPromptBuilder(template)

    def test_run_without_template(self):
        prompt_builder = ChatPromptBuilder()
        with pytest.raises(
            ValueError, match="The ChatPromptBuilder requires a non-empty list of ChatMessage instances"
        ):
            prompt_builder.run()

    def test_run_with_empty_chat_message_list(self):
        prompt_builder = ChatPromptBuilder(template=[], variables=["documents"])
        with pytest.raises(
            ValueError, match="The ChatPromptBuilder requires a non-empty list of ChatMessage instances"
        ):
            prompt_builder.run()

    def test_chat_message_list_with_mixed_object_list(self):
        prompt_builder = ChatPromptBuilder(
            template=[ChatMessage.from_user("Hello"), "there world"], variables=["documents"]
        )
        with pytest.raises(
            ValueError, match="The ChatPromptBuilder expects a list containing only ChatMessage instances"
        ):
            prompt_builder.run()

    def test_provided_template_variables(self):
        prompt_builder = ChatPromptBuilder(variables=["documents"], required_variables=["city"])

        # both variables are provided
        prompt_builder._validate_variables({"name", "city"})

        # provided variables are a superset of the required variables
        prompt_builder._validate_variables({"name", "city", "age"})

        with pytest.raises(ValueError):
            prompt_builder._validate_variables({"name"})

    def test_example_in_pipeline(self):
        default_template = [
            ChatMessage.from_user("Here is the document: {{documents[0].content}} \\n Answer: {{query}}")
        ]
        prompt_builder = ChatPromptBuilder(template=default_template, variables=["documents"])

        @component
        class DocumentProducer:
            @component.output_types(documents=list[Document])
            def run(self, doc_input: str):
                return {"documents": [Document(content=doc_input)]}

        pipe = Pipeline()
        pipe.add_component("doc_producer", DocumentProducer())
        pipe.add_component("prompt_builder", prompt_builder)
        pipe.connect("doc_producer.documents", "prompt_builder.documents")

        template = [ChatMessage.from_user("Here is the document: {{documents[0].content}} \n Query: {{query}}")]
        result = pipe.run(
            data={
                "doc_producer": {"doc_input": "Hello world, I live in Berlin"},
                "prompt_builder": {
                    "template": template,
                    "template_variables": {"query": "Where does the speaker live?"},
                },
            }
        )

        assert result == {
            "prompt_builder": {
                "prompt": [
                    ChatMessage.from_user(
                        "Here is the document: Hello world, I live in Berlin \n Query: Where does the speaker live?"
                    )
                ]
            }
        }

    def test_example_in_pipeline_simple(self):
        default_template = [ChatMessage.from_user("This is the default prompt:\n Query: {{query}}")]
        prompt_builder = ChatPromptBuilder(template=default_template)

        pipe = Pipeline()
        pipe.add_component("prompt_builder", prompt_builder)

        # using the default prompt
        result = pipe.run(data={"query": "Where does the speaker live?"})
        expected_default = {
            "prompt_builder": {
                "prompt": [ChatMessage.from_user("This is the default prompt:\n Query: Where does the speaker live?")]
            }
        }
        assert result == expected_default

        # using the dynamic prompt
        result = pipe.run(
            data={
                "query": "Where does the speaker live?",
                "template": [ChatMessage.from_user("This is the dynamic prompt:\n Query: {{query}}")],
            }
        )
        expected_dynamic = {
            "prompt_builder": {
                "prompt": [ChatMessage.from_user("This is the dynamic prompt:\n Query: Where does the speaker live?")]
            }
        }
        assert result == expected_dynamic

    def test_warning_no_required_variables(self, caplog):
        with caplog.at_level(logging.WARNING):
            _ = ChatPromptBuilder(
                template=[
                    ChatMessage.from_system("Write your response in this language:{{language}}"),
                    ChatMessage.from_user("Tell me about {{location}}"),
                ]
            )
            assert "ChatPromptBuilder has 2 prompt variables, but `required_variables` is not set. " in caplog.text

    def test_with_custom_dateformat(self) -> None:
        template = [ChatMessage.from_user("Formatted date: {% now 'UTC', '%Y-%m-%d' %}")]
        builder = ChatPromptBuilder(template=template)

        result = builder.run()["prompt"]

        now_formatted = f"Formatted date: {arrow.now('UTC').strftime('%Y-%m-%d')}"
        assert len(result) == 1
        assert result[0].role == "user"
        assert result[0].text == now_formatted

    def test_with_different_timezone(self) -> None:
        template = [ChatMessage.from_user("Current time in New York is: {% now 'America/New_York' %}")]
        builder = ChatPromptBuilder(template=template)

        result = builder.run()["prompt"]

        now_ny = f"Current time in New York is: {arrow.now('America/New_York').strftime('%Y-%m-%d %H:%M:%S')}"
        assert len(result) == 1
        assert result[0].role == "user"
        assert result[0].text == now_ny

    def test_date_with_addition_offset(self) -> None:
        template = [ChatMessage.from_user("Time after 2 hours is: {% now 'UTC' + 'hours=2' %}")]
        builder = ChatPromptBuilder(template=template)

        result = builder.run()["prompt"]

        now_plus_2 = f"Time after 2 hours is: {(arrow.now('UTC').shift(hours=+2)).strftime('%Y-%m-%d %H:%M:%S')}"
        assert len(result) == 1
        assert result[0].role == "user"
        assert result[0].text == now_plus_2

    def test_date_with_subtraction_offset(self) -> None:
        template = [ChatMessage.from_user("Time after 12 days is: {% now 'UTC' - 'days=12' %}")]
        builder = ChatPromptBuilder(template=template)

        result = builder.run()["prompt"]

        now_minus_12 = f"Time after 12 days is: {(arrow.now('UTC').shift(days=-12)).strftime('%Y-%m-%d %H:%M:%S')}"
        assert len(result) == 1
        assert result[0].role == "user"
        assert result[0].text == now_minus_12

    def test_invalid_timezone(self) -> None:
        template = [ChatMessage.from_user("Current time is: {% now 'Invalid/Timezone' %}")]
        builder = ChatPromptBuilder(template=template)

        # Expect ValueError for invalid timezone
        with pytest.raises(ValueError, match="Invalid timezone"):
            builder.run()

    def test_invalid_offset(self) -> None:
        template = [ChatMessage.from_user("Time after invalid offset is: {% now 'UTC' + 'invalid_offset' %}")]
        builder = ChatPromptBuilder(template=template)

        # Expect ValueError for invalid offset
        with pytest.raises(ValueError, match="Invalid offset or operator"):
            builder.run()

    def test_multiple_messages_with_date_template(self) -> None:
        template = [
            ChatMessage.from_user("Current date is: {% now 'UTC' %}"),
            ChatMessage.from_assistant("Thank you for providing the date"),
            ChatMessage.from_user("Yesterday was: {% now 'UTC' - 'days=1' %}"),
        ]
        builder = ChatPromptBuilder(template=template)

        result = builder.run()["prompt"]

        now = f"Current date is: {arrow.now('UTC').strftime('%Y-%m-%d %H:%M:%S')}"
        yesterday = f"Yesterday was: {(arrow.now('UTC').shift(days=-1)).strftime('%Y-%m-%d %H:%M:%S')}"

        assert len(result) == 3
        assert result[0].role == "user"
        assert result[0].text == now
        assert result[1].role == "assistant"
        assert result[1].text == "Thank you for providing the date"
        assert result[2].role == "user"
        assert result[2].text == yesterday


class TestChatPromptBuilderDynamic:
    def test_multiple_templated_chat_messages(self):
        prompt_builder = ChatPromptBuilder()
        language = "French"
        location = "Berlin"
        messages = [
            ChatMessage.from_system("Write your response in this language:{{language}}"),
            ChatMessage.from_user("Tell me about {{location}}"),
        ]

        result = prompt_builder.run(template_variables={"language": language, "location": location}, template=messages)
        assert result["prompt"] == [
            ChatMessage.from_system("Write your response in this language:French"),
            ChatMessage.from_user("Tell me about Berlin"),
        ], "The templated messages should match the expected output."

    def test_multiple_templated_chat_messages_in_place(self):
        prompt_builder = ChatPromptBuilder()
        language = "French"
        location = "Berlin"
        messages = [
            ChatMessage.from_system("Write your response ins this language:{{language}}"),
            ChatMessage.from_user("Tell me about {{location}}"),
        ]

        res = prompt_builder.run(template_variables={"language": language, "location": location}, template=messages)
        assert res == {
            "prompt": [
                ChatMessage.from_system("Write your response ins this language:French"),
                ChatMessage.from_user("Tell me about Berlin"),
            ]
        }, "The templated messages should match the expected output."

    def test_some_templated_chat_messages(self):
        prompt_builder = ChatPromptBuilder()
        language = "English"
        location = "Paris"
        messages = [
            ChatMessage.from_system("Please, respond in the following language: {{language}}."),
            ChatMessage.from_user("I would like to learn more about {{location}}."),
            ChatMessage.from_assistant("Yes, I can help you with that {{subject}}"),
            ChatMessage.from_user("Ok so do so please, be elaborate."),
        ]

        result = prompt_builder.run(template_variables={"language": language, "location": location}, template=messages)

        expected_messages = [
            ChatMessage.from_system("Please, respond in the following language: English."),
            ChatMessage.from_user("I would like to learn more about Paris."),
            ChatMessage.from_assistant(
                "Yes, I can help you with that {{subject}}"
            ),  # assistant message should not be templated
            ChatMessage.from_user("Ok so do so please, be elaborate."),
        ]

        assert result["prompt"] == expected_messages, "The templated messages should match the expected output."

    def test_example_in_pipeline(self):
        prompt_builder = ChatPromptBuilder()

        pipe = Pipeline()
        pipe.add_component("prompt_builder", prompt_builder)

        location = "Berlin"
        system_message = ChatMessage.from_system(
            "You are a helpful assistant giving out valuable information to tourists."
        )
        messages = [system_message, ChatMessage.from_user("Tell me about {{location}}")]

        res = pipe.run(data={"prompt_builder": {"template_variables": {"location": location}, "template": messages}})
        assert res == {
            "prompt_builder": {
                "prompt": [
                    ChatMessage.from_system("You are a helpful assistant giving out valuable information to tourists."),
                    ChatMessage.from_user("Tell me about Berlin"),
                ]
            }
        }

        messages = [
            system_message,
            ChatMessage.from_user("What's the weather forecast for {{location}} in the next {{day_count}} days?"),
        ]

        res = pipe.run(
            data={
                "prompt_builder": {"template_variables": {"location": location, "day_count": "5"}, "template": messages}
            }
        )
        assert res == {
            "prompt_builder": {
                "prompt": [
                    ChatMessage.from_system("You are a helpful assistant giving out valuable information to tourists."),
                    ChatMessage.from_user("What's the weather forecast for Berlin in the next 5 days?"),
                ]
            }
        }

    def test_example_in_pipeline_with_multiple_templated_messages(self):
        # no parameter init, we don't use any runtime template variables
        prompt_builder = ChatPromptBuilder()

        pipe = Pipeline()
        pipe.add_component("prompt_builder", prompt_builder)

        location = "Berlin"
        system_message = ChatMessage.from_system(
            "You are a helpful assistant giving out valuable information to tourists in {{language}}."
        )
        messages = [system_message, ChatMessage.from_user("Tell me about {{location}}")]

        res = pipe.run(
            data={
                "prompt_builder": {
                    "template_variables": {"location": location, "language": "German"},
                    "template": messages,
                }
            }
        )
        assert res == {
            "prompt_builder": {
                "prompt": [
                    ChatMessage.from_system(
                        "You are a helpful assistant giving out valuable information to tourists in German."
                    ),
                    ChatMessage.from_user("Tell me about Berlin"),
                ]
            }
        }

        messages = [
            system_message,
            ChatMessage.from_user("What's the weather forecast for {{location}} in the next {{day_count}} days?"),
        ]

        res = pipe.run(
            data={
                "prompt_builder": {
                    "template_variables": {"location": location, "day_count": "5", "language": "English"},
                    "template": messages,
                }
            }
        )
        assert res == {
            "prompt_builder": {
                "prompt": [
                    ChatMessage.from_system(
                        "You are a helpful assistant giving out valuable information to tourists in English."
                    ),
                    ChatMessage.from_user("What's the weather forecast for Berlin in the next 5 days?"),
                ]
            }
        }

    def test_pipeline_complex(self):
        @component
        class ValueProducer:
            def __init__(self, value_to_produce: str):
                self.value_to_produce = value_to_produce

            @component.output_types(value_output=str)
            def run(self):
                return {"value_output": self.value_to_produce}

        pipe = Pipeline()
        pipe.add_component("prompt_builder", ChatPromptBuilder(variables=["value_output"]))
        pipe.add_component("value_producer", ValueProducer(value_to_produce="Berlin"))
        pipe.connect("value_producer.value_output", "prompt_builder")

        messages = [
            ChatMessage.from_system("You give valuable information to tourists."),
            ChatMessage.from_user("Tell me about {{value_output}}"),
        ]

        res = pipe.run(data={"template": messages})
        assert res == {
            "prompt_builder": {
                "prompt": [
                    ChatMessage.from_system("You give valuable information to tourists."),
                    ChatMessage.from_user("Tell me about Berlin"),
                ]
            }
        }

    def test_to_dict(self):
        comp = ChatPromptBuilder(
            template=[ChatMessage.from_user("text and {var}"), ChatMessage.from_assistant("content {required_var}")],
            variables=["var", "required_var"],
            required_variables=["required_var"],
        )

        assert comp.to_dict() == {
            "type": "haystack.components.builders.chat_prompt_builder.ChatPromptBuilder",
            "init_parameters": {
                "template": [
                    {"content": [{"text": "text and {var}"}], "role": "user", "meta": {}, "name": None},
                    {"content": [{"text": "content {required_var}"}], "role": "assistant", "meta": {}, "name": None},
                ],
                "variables": ["var", "required_var"],
                "required_variables": ["required_var"],
            },
        }

    def test_from_dict(self):
        comp = ChatPromptBuilder.from_dict(
            data={
                "type": "haystack.components.builders.chat_prompt_builder.ChatPromptBuilder",
                "init_parameters": {
                    "template": [
                        {"content": [{"text": "text and {var}"}], "role": "user", "meta": {}, "name": None},
                        {
                            "content": [{"text": "content {required_var}"}],
                            "role": "assistant",
                            "meta": {},
                            "name": None,
                        },
                    ],
                    "variables": ["var", "required_var"],
                    "required_variables": ["required_var"],
                },
            }
        )

        assert comp.template == [
            ChatMessage.from_user("text and {var}"),
            ChatMessage.from_assistant("content {required_var}"),
        ]
        assert comp._variables == ["var", "required_var"]
        assert comp._required_variables == ["required_var"]

    def test_from_dict_template_none(self):
        comp = ChatPromptBuilder.from_dict(
            data={
                "type": "haystack.components.builders.chat_prompt_builder.ChatPromptBuilder",
                "init_parameters": {"template": None},
            }
        )

        assert comp.template is None
        assert comp._variables is None
        assert comp._required_variables is None

    def test_chat_message_list_with_templatize_part_init_raises_error(self):
        template = [ChatMessage.from_user("This is a {{ variable | templatize_part }}")]
        with pytest.raises(ValueError, match="templatize_part filter cannot be used"):
            ChatPromptBuilder(template=template)

    def test_chat_message_list_with_templatize_part_run_raises_error(self):
        builder = ChatPromptBuilder()
        template = [ChatMessage.from_user("This is a {{ variable | templatize_part }}")]
        with pytest.raises(ValueError, match="templatize_part filter cannot be used"):
            builder.run(template=template, variable="test")


class TestChatPromptBuilderWithStrTemplate:
    def test_init(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)

        assert builder.template == template
        assert builder._variables is None
        assert builder._required_variables is None
        assert builder.variables == ["name"]

    def test_init_with_invalid_template(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}!
        {% endmessage %}
        """
        with pytest.raises(TemplateSyntaxError):
            ChatPromptBuilder(template=template)

    def test_run(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run(name="John")
        assert result["prompt"] == [ChatMessage.from_user("Hello, my name is John!")]

    def test_run_template_variable(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run(template_variables={"name": "John"})
        assert result["prompt"] == [ChatMessage.from_user("Hello, my name is John!")]

    def test_run_template_variable_overrides_variable(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run(name="John", template_variables={"name": "Jane"})
        assert result["prompt"] == [ChatMessage.from_user("Hello, my name is Jane!")]

    def test_run_without_input(self):
        template = """
        {% message role="user" %}
        Hello, my name is Lukas!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run()
        assert result["prompt"] == [ChatMessage.from_user("Hello, my name is Lukas!")]

    def test_run_with_missing_input(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run()
        assert result["prompt"] == [ChatMessage.from_user("Hello, my name is !")]

    def test_run_with_missing_required_input(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template, required_variables=["name"])
        with pytest.raises(ValueError):
            builder.run()

    def test_run_with_missing_required_input_using_star(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template, required_variables="*")
        with pytest.raises(ValueError):
            builder.run()

    def test_run_with_variables_and_runtime_template(self):
        variables = ["name"]

        builder = ChatPromptBuilder(variables=variables)

        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        template_variables = {"name": "John"}

        expected_result = {"prompt": [ChatMessage.from_user("Hello, my name is John!")]}

        assert builder.run(template_variables=template_variables, template=template) == expected_result

    def test_run_overwriting_default_template(self):
        initial_template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=initial_template)

        runtime_template = """
        {% message role="user" %}
        Hello, I come from {{country}}!
        {% endmessage %}
        """
        result = builder.run(template_variables={"country": "Italy"}, template=runtime_template)
        assert result["prompt"] == [ChatMessage.from_user("Hello, I come from Italy!")]

    def test_run_with_name_and_meta(self):
        template = """
        {% message role="user" name="John" meta={"key": "value"} %}
        Hello from {{country}}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run(country="Italy")
        assert result["prompt"] == [ChatMessage.from_user("Hello from Italy!", name="John", meta={"key": "value"})]

    def test_multiline_template(self):
        template = """
{% message role="user" %}
Hello, my name is {{name}}!
Second line.
Third line.
{% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run(name="John")
        assert result["prompt"] == [ChatMessage.from_user("Hello, my name is John!\nSecond line.\nThird line.")]

    def test_with_now_filter(self):
        template = """
        {% message role="user" %}
        Hello, the date is {% now 'UTC', '%Y-%m-%d'%}!
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run()

        expected_date = arrow.now("UTC").strftime("%Y-%m-%d")
        assert result["prompt"] == [ChatMessage.from_user(f"Hello, the date is {expected_date}!")]

    def test_run_multiple_messages(self):
        template = """
        {% message role="system" %}
        You are a {{adjective}} assistant.
        {% endmessage %}

        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}

        {% message role="assistant" %}
        Hello, {{name}}! How can I help you today?
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        result = builder.run(name="John", adjective="helpful")
        assert result["prompt"] == [
            ChatMessage.from_system("You are a helpful assistant."),
            ChatMessage.from_user("Hello, my name is John!"),
            ChatMessage.from_assistant("Hello, John! How can I help you today?"),
        ]

    def test_run_multiple_images(self, base64_image_string):
        template = """
        {% message role="user" %}
        Hello! I am {{user_name}}. What's the difference between the following images?
        {% for image in images %}
        {{ image | templatize_part }}
        {% endfor %}
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        images = [
            ImageContent(base64_image=base64_image_string, mime_type="image/png"),
            ImageContent(base64_image=base64_image_string, mime_type="image/png"),
        ]
        result = builder.run(user_name="John", images=images)

        assert result["prompt"] == [
            ChatMessage.from_user(
                content_parts=["Hello! I am John. What's the difference between the following images?", *images]
            )
        ]

    def test_run_reasoning(self):
        template = """
        {% message role="user" %}
        Hello! I am {{user_name}}. How much is 2 + 2?
        {% endmessage %}

        {% message role="assistant" %}
        {{ reasoning | templatize_part }}
        The answer is 4.
        {% endmessage %}
        """
        builder = ChatPromptBuilder(template=template)
        reasoning = ReasoningContent(reasoning_text="Let me think about it...", extra={"key": "value"})
        result = builder.run(user_name="John", reasoning=reasoning)

        assert result["prompt"] == [
            ChatMessage.from_user(text="Hello! I am John. How much is 2 + 2?"),
            ChatMessage.from_assistant(reasoning=reasoning, text="The answer is 4."),
        ]

    def test_to_dict(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}

        {% message role="assistant" %}
        Hello, I am {{assistant_name}}! How can I help you today?
        {% endmessage %}
        """
        builder = ChatPromptBuilder(
            template=template, variables=["name", "assistant_name"], required_variables=["name"]
        )

        assert builder.to_dict() == {
            "type": "haystack.components.builders.chat_prompt_builder.ChatPromptBuilder",
            "init_parameters": {
                "template": template,
                "variables": ["name", "assistant_name"],
                "required_variables": ["name"],
            },
        }

    def test_from_dict(self):
        template = """
        {% message role="user" %}
        Hello, my name is {{name}}!
        {% endmessage %}

        {% message role="assistant" %}
        Hello, I am {{assistant_name}}! How can I help you today?
        {% endmessage %}
        """

        data = {
            "type": "haystack.components.builders.chat_prompt_builder.ChatPromptBuilder",
            "init_parameters": {
                "template": template,
                "variables": ["name", "assistant_name"],
                "required_variables": ["name"],
            },
        }
        builder = ChatPromptBuilder.from_dict(data)
        assert builder.template == template
        assert builder.variables == ["name", "assistant_name"]
        assert builder.required_variables == ["name"]

    def test_variables_correct_with_assignment(self):
        template = """{% message role="user" %}
{% if existing_documents is not none -%}
{% set x = existing_documents|length -%}
{% else -%}
{% set x = 0 -%}
{% endif -%}
The number is {{ x }}!
{% endmessage %}
"""
        builder = ChatPromptBuilder(template=template, required_variables="*")
        assert builder.variables == ["existing_documents"]
        assert builder.required_variables == "*"
        res = builder.run(existing_documents=None)
        assert res["prompt"][0].text == "The number is 0!"

    def test_variables_correct_with_tuple_assignment(self):
        template = """{% message role="user" %}
{% if name is not none -%}
{% set x, y = (0, 1) %}
{% else -%}
{% set x, y = (2, 3) %}
{% endif -%}
x={{ x }}, y={{ y }}
Hello, my name is {{name}}!
{% endmessage %}
"""
        builder = ChatPromptBuilder(template=template, required_variables="*")
        assert builder.variables == ["name"]
        assert builder.required_variables == "*"
        res = builder.run(name="John")
        assert res["prompt"][0].text == "x=0, y=1\nHello, my name is John!"

    def test_variables_correct_with_list_assignment(self):
        template = """{% message role="user" %}
{% if name is not none -%}
{% set x, y = [0, 1] %}
{% else -%}
{% set x, y = [2, 3] %}
{% endif -%}
x={{ x }}, y={{ y }}
Hello, my name is {{name}}!
{% endmessage %}
"""
        builder = ChatPromptBuilder(template=template, required_variables="*")
        assert builder.variables == ["name"]
        assert builder.required_variables == "*"
        res = builder.run(name="John")
        assert res["prompt"][0].text == "x=0, y=1\nHello, my name is John!"
